home *** CD-ROM | disk | FTP | other *** search
- /* Copyright (C) 1991, 1992, 1993, 1994 Free Software Foundation, Inc.
- This file is part of the GNU C Library.
-
- The GNU C Library is free software; you can redistribute it and/or
- modify it under the terms of the GNU Library General Public License as
- published by the Free Software Foundation; either version 2 of the
- License, or (at your option) any later version.
-
- The GNU C Library is distributed in the hope that it will be useful,
- but WITHOUT ANY WARRANTY; without even the implied warranty of
- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- Library General Public License for more details.
-
- You should have received a copy of the GNU Library General Public
- License along with the GNU C Library; see the file COPYING.LIB. If
- not, write to the Free Software Foundation, Inc., 675 Mass Ave,
- Cambridge, MA 02139, USA. */
-
- #include <ansidecl.h>
- #include <errno.h>
- #include <stdio.h>
- #include <string.h>
-
-
- /* Write NMEMB chunks of SIZE bytes each from PTR onto STREAM. */
- size_t
- DEFUN(fwrite, (ptr, size, nmemb, stream),
- CONST PTR ptr AND size_t size AND
- size_t nmemb AND register FILE *stream)
- {
- register CONST unsigned char *p = (CONST unsigned char *) ptr;
- register size_t to_write = size * nmemb;
- register size_t written = 0;
- int newlinep;
- size_t buffer_space;
- int default_func;
-
- if (!__validfp (stream) || !stream->__mode.__write)
- {
- errno = EINVAL;
- return 0;
- }
-
- if (ferror (stream))
- return 0;
- if (p == NULL || to_write == 0)
- return 0;
-
- if (!stream->__seen || stream->__put_limit == stream->__buffer)
- {
- /* This stream has never been seen before.
- Calling __flshfp will give it a buffer
- and I/O functions if it needs them. */
- if (__flshfp (stream, *p++) == EOF)
- return 0;
- if (--to_write == 0)
- return 1;
- else
- ++written;
- }
-
- default_func
- = stream->__room_funcs.__output == __default_room_functions.__output;
-
- {
- int save = errno;
-
- if (__stdio_check_offset (stream) == EOF && errno != ESPIPE)
- {
- stream->__error = 1;
- goto done;
- }
-
- errno = save;
- }
-
- if (stream->__buffer == NULL && default_func &&
- stream->__offset == stream->__target)
- write_through:
- /* This is an unbuffered stream using the standard output
- buffer-flushing function, so we just do a straight write. */
- {
- int count = (stream->__io_funcs.__write == NULL ? to_write :
- (*stream->__io_funcs.__write) (stream->__cookie,
- (CONST char *) p,
- to_write));
- if (count > 0)
- {
- written += count;
- if (stream->__offset != -1)
- {
- stream->__offset += count;
- stream->__target = stream->__offset;
- }
- to_write -= count;
- p += count;
- }
- else
- stream->__error = 1;
- goto done;
- }
-
- /* We ignore the end pointer here since we want to find out how much space
- is really in the buffer, even for a line-buffered stream. */
- buffer_space = stream->__bufsize - (stream->__bufp - stream->__buffer);
-
- newlinep = (stream->__linebuf &&
- memchr ((CONST PTR) p, '\n', to_write) != NULL);
-
- if (newlinep && stream->__bufp == stream->__buffer &&
- stream->__offset == stream->__target)
- /* The buffer's empty, and we want to write our data
- out soon anyway, so just write it straight out. */
- goto write_through;
-
- if (stream->__bufsize == 0 && !default_func)
- {
- /* No buffer, and a special function.
- We can't do much better than putc. */
- while (to_write-- > 0)
- {
- if (__flshfp (stream, *p++) == EOF)
- break;
- else
- ++written;
- }
- }
- else if (!default_func || buffer_space >= to_write)
- fill_buffer:
- /* There is enough room in the buffer for everything we
- want to write or the user has specified his own output
- buffer-flushing/expanding function. */
- while (to_write > 0)
- {
- register size_t n = to_write;
-
- if (n > buffer_space)
- n = buffer_space;
-
- buffer_space -= n;
-
- written += n;
- to_write -= n;
-
- if (n < 20)
- while (n-- > 0)
- *stream->__bufp++ = *p++;
- else
- {
- memcpy ((PTR) stream->__bufp, (PTR) p, n);
- stream->__bufp += n;
- p += n;
- }
-
- if (buffer_space == 0 || (to_write == 0 && newlinep))
- {
- /* We've filled the buffer, so flush it. */
- if (fflush (stream) == EOF)
- break;
- /* Reset our record of the space available in the buffer,
- since we have just flushed it. */
- buffer_space = (stream->__bufsize -
- (stream->__bufp - stream->__buffer));
- }
- }
- else
- {
- /* It won't all fit in the buffer. */
-
- if (stream->__bufp != stream->__buffer)
- {
- /* There are characters in the buffer. Flush them. */
- if (__flshfp (stream, EOF) == EOF)
- goto done;
- }
-
- /* The buffer has been flushed.
- Now either fill it or write directly. */
-
- buffer_space = stream->__bufsize - (stream->__bufp - stream->__buffer);
-
- if (stream->__offset == stream->__target &&
- (buffer_space < to_write || newlinep))
- /* What we have to write is bigger than the buffer,
- or it contains a newline and we're line-buffered,
- so write it out. */
- goto write_through;
- else
- /* It will fit in the buffer. */
- goto fill_buffer;
- }
-
- done:;
- return (size_t) written / size;
- }
-